inst++; //skip ModR/M byte
if ( ad_size != WORD && mod != 3 && rm == 4 ) {
- rm = *inst & 7;
inst++; //skip SIB byte
}
return val;
}
+/* Some instructions, like "add $imm8, r/m16"/"MOV $imm32, r/m64" require
+ * the src immediate operand be sign-extented befere the op is executed. Here
+ * we always sign-extend the operand to a "unsigned long" variable.
+ *
+ * Note: to simplify the logic here, the sign-extension here may be performed
+ * redundantly against some instructions, like "MOV $imm16, r/m16" -- however
+ * this is harmless, since we always remember the operand's size.
+ */
+static inline unsigned long get_immediate_sign_ext(int ad_size,
+ const unsigned char *inst,
+ int op_size)
+{
+ unsigned long result = get_immediate(ad_size, inst, op_size);
+
+ if ( op_size == QUAD )
+ op_size = LONG;
+
+ ASSERT( op_size == BYTE || op_size == WORD || op_size == LONG );
+
+ if ( result & (1UL << ((8*op_size) - 1)) )
+ {
+ unsigned long mask = ~0UL >> (8 * (sizeof(mask) - op_size));
+ result = ~mask | (result & mask);
+ }
+ return result;
+}
+
static inline int get_index(const unsigned char *inst, unsigned char rex)
{
int mod, reg, rm;
case 8:
if ( *op_size == 0 )
*op_size = rex & 0x8 ? QUAD : LONG;
- if ( *ad_size == 0 )
+ if ( *ad_size == WORD )
+ *ad_size = LONG;
+ else if ( *ad_size == 0 )
*ad_size = QUAD;
break;
#endif
/* opcode 0x83 always has a single byte operand */
if ( opcode[0] == 0x83 )
mmio_op->immediate =
- (signed char)get_immediate(*ad_size, opcode + 1, BYTE);
+ get_immediate_sign_ext(*ad_size, opcode + 1, BYTE);
else
mmio_op->immediate =
- get_immediate(*ad_size, opcode + 1, *op_size);
+ get_immediate_sign_ext(*ad_size, opcode + 1, *op_size);
mmio_op->operand[0] = mk_operand(size_reg, 0, 0, IMMEDIATE);
mmio_op->operand[1] = mk_operand(size_reg, 0, 0, MEMORY);
mmio_op->operand[0] = mk_operand(*op_size, 0, 0, IMMEDIATE);
mmio_op->immediate =
- get_immediate(*ad_size, opcode + 1, *op_size);
+ get_immediate_sign_ext(*ad_size, opcode + 1, *op_size);
mmio_op->operand[1] = mk_operand(*op_size, 0, 0, MEMORY);
return DECODE_success;
mmio_op->operand[0] = mk_operand(size_reg, 0, 0, IMMEDIATE);
mmio_op->immediate =
- get_immediate(*ad_size, opcode + 1, *op_size);
+ get_immediate_sign_ext(*ad_size, opcode + 1, *op_size);
mmio_op->operand[1] = mk_operand(size_reg, 0, 0, MEMORY);
return DECODE_success;